import random
import re
from datetime import datetime

from flask import current_app
from flask import json, jsonify
from flask import make_response
from flask import request
from flask import session

from info import constants, db
from info import redis_store
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.captcha.captcha import captcha
from info.utils.response_code import RET
from . import passport_blue

#功能描述: 退出
# 请求路径: /passport/logout
# 请求方式: POST
# 请求参数: 无
# 返回值: errno, errmsg
# session["user_id"] = user.id
# session["user_name"] = user.nick_name
# session["mobile"] = user.mobile

@passport_blue.route('/logout', methods=['POST'])
def logout():

    session.pop("user_id",None)
    session.pop("user_name",None)
    session.pop("mobile",None)
    session.pop("is_admin",None)

    return jsonify(errno=RET.OK,errmsg="退出成功")



#功能描述: 用户登陆
# 请求路径: /passport/login
# 请求方式: POST
# 请求参数: mobile,password
# 返回值: errno, errmsg
@passport_blue.route('/login', methods=['POST'])
def login():
    """
    思路分析:
    1.获取参数

    2.校验参数,为空校验

    3.通过手机号,去数据库查询这个用户对象

    4.判断该用户是否存在

    5.校验用户密码是否正确

    6.保存用户的登陆状态到session

    7.返回响应
    :return:
    """
    """

     var params = {
            "mobile":mobile,
            "password":password
        }

    """
    # 1.获取参数
    mobile=request.json.get("mobile")
    password=request.json.get("password")
    # 2.校验参数,为空校验
    if not all([mobile,password]):
        return jsonify(errno=RET.PARAMERR, errmsg="参数不全")

    # 3.通过手机号,去数据库查询这个用户对象
    try:
       user=User.query.filter(User.mobile == mobile).first()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="查询用户异常")
    # 4.判断该用户是否存在
    if not user:
        return jsonify(errno=RET.NODATA, errmsg="用户不存在")

    # 5.校验用户密码是否正确
    if not user.check_passowrd(password):
        return jsonify(errno=RET.DATAERR, errmsg="密码错误")

    # 6.保存用户的登陆状态到session
    session["user_id"]=user.id
    session["user_name"]=user.nick_name
    session["mobile"]=user.mobile


    user.last_login=datetime.now()

    # 7.返回响应
    return jsonify(errno=RET.OK,errmsg="登录成功")








#功能描述: 注册用户
# 请求路径: /passport/register
# 请求方式: POST
# 请求参数: mobile, sms_code,password
# 返回值: errno, errmsg

@passport_blue.route('/register', methods=['POST'])
def register():
    """
       1.获取参数
       2.校验参数,为空校验
       3.手机号格式校验
       4.根据手机号,去redis中取出短信验证码
       5.判断短信验证码是否过期
       6.删除redis的短信验证码
       7.判断传入的短信验证码和redis中取出的是否一致
       8.创建用户对象,设置属性
       9.保存用户到数据库mysql
       10.返回响应
       :return:
       """
    # 1.获取参数
    # 1.获取参数
    # json_data = request.data
    # dict_data = json.loads(json_data)

    # 上面两句话可以写成一句话
    # request.get_json() #等价于request.json
    """

     var params = {
            "mobile":mobile,
            "sms_code":smscode,
            "password":password
        }

    """
    dict_data=request.json
    mobile=dict_data.get("mobile")
    smscode=dict_data.get("sms_code")
    password=dict_data.get("password")

    # 2.校验参数,为空校验
    if not all([mobile,sms_code,password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

    # 3.手机号格式校验
    if not re.match("1[356789]\d{9}$",mobile):
        return jsonify(errno=RET.DATAERR, errmsg="手机号格式不正确")

    # 4.根据手机号,去redis中取出短信验证码
    try:
        redis_sms_code=redis_store.get("sms_code:%s"%mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="获取短信验证码异常")

    # 5.判断短信验证码是否过期
    if not redis_sms_code:
        return jsonify(errno=RET.NODATA, errmsg="短信验证码已过期")

    # 6.删除redis的短信验证码
    try:
        redis_store.delete("sms_code:%s"%mobile)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg="删除短信验证码异常")

    # 7.判断传入的短信验证码和redis中取出的是否一致
    if smscode !=redis_sms_code:
        return jsonify(errno=RET.DATAERR, errmsg="短信验证码填写有误")

    # 8.创建用户对象,设置属性
    user=User()
    user.nick_name=mobile
    user.password = password
    user.mobile=mobile

    # 9.保存用户到数据库mysql
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()#回滚
        return jsonify(errno=RET.DBERR, errmsg="注册用户失败")
    return jsonify(errno=RET.OK,errmsg="注册成功")

# 功能:发送短信验证码
# 请求路径: /passport/sms_code
# 请求方式: POST
# 请求参数: mobile, image_code(验证码),image_code_id(编号)
# 返回值: errno, errmsg
@passport_blue.route('/sms_code',methods=["POST"])
def sms_code():
    """
        思路分析:
        1.获取参数
        2.校验参数,为空校验
        3.校验手机号格式是否正确
        4.通过验证码编号取出,redis中的图片验证码A
        5.判断验证码A是否过期
        6.删除,redis中的图片验证码
        7.判断验证码A和传入进来的图片验证码是否相等
        8.生成短信验证码
        9.发送短信验证码,调用ccp方法
        10.存储短信验证码到redis中
        11.返回发送状态
        :return:
        """
    # 1.获取参数
    # json_data=request.data.decode()
    # dict_data=json.load(json_data)
    dict_data=request.json
    """
     var params = {
        "mobile":mobile,
        "image_code":imageCode,
        "image_code_id":imageCodeId
    }

    """
    mobile=dict_data.get("mobile")
    image_code = dict_data.get("image_code")
    image_code_id = dict_data.get("image_code_id")
    # 2.校验参数,为空校验
    if not all([mobile,image_code,image_code_id]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不完整")

    # 3.校验手机号格式是否正确
    if not re.match("1[35789]\d{9}$",mobile):
        return jsonify(errno=RET.DATAERR,errmsg="手机号格式错误")
    #4.通过验证码编号取出,redis中的图片验证码A
    try:
        redis_image_code=redis_store.get("image_code:%s"%image_code_id)

    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取图片验证码异常")
    # 5.判断验证码A是否过期
    if not redis_image_code:
        return jsonify(errno=RET.NODATA,errmsg="图片验证码已过期")
    # 6.删除,redis中的图片验证码
    try:
        redis_store.delete("image_code:%s"%image_code_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="图片验证码操作异常")

    # 7.判断验证码A和传入进来的图片验证码是否相等
    if redis_image_code.lower()!=image_code.lower():
        return jsonify(errno=RET.DATAERR,errmsg="图片验证码填写错误")
    # 8.生成短信验证码
    sms_code='%06d'%random.randint(0,999999)
    current_app.logger.debug("短信验证码：=%s"%sms_code)


    # ccp=CCP()
    # try:
    #     result=ccp.send_template_sms(mobile,[sms_code,constants.SMS_CODE_REDIS_EXPIRES/60],1)
    # except Exception as e:
    #     current_app.logger.error(e)
    #     return jsonify(errno=RET.THIRDERR,errmsg="云通讯发送异常")
    #
    # if result== -1:
    #     return jsonify(errno=RET.DATAERR,errmsg="短信发送失败")

    # 10.存储短信验证码到redis中
    try:
        redis_store.set("sms_code:%s"%mobile,sms_code,constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="保存图片验证码失败")

    return jsonify(errno=RET.OK,errmsg="发送成功",data=sms_code)


#获取返回,一张图片
# 请求路径: /passport/image_code
# 请求方式: GET
# 请求参数: cur_id, pre_id
# 返回值: 图片验证码


@passport_blue.route('/image_code')
def image_code():
    """
          思路分析:
          1.获取参数
          2.校验参数(为空校验)
          3.生成图片验证码
          4.保存图片验证码到redis
          5.判断是否有上个图片验证码编号,有则删除
          6.返回图片验证码即可
          :return:
          """
    # 1.获取参数
    cur_id = request.args.get('cur_id')
    pre_id = request.args.get('pre_id')
    # 2.校验参数(为空校验)
    if not cur_id:
        return jsonify(errno=RET.PARAMERR,errmsg="图片验证码编号不存在")
    # 3.生成图片验证码
    name,text,image_data=captcha.generate_captcha()
    try:
        # 4.保存图片验证码到redis
        redis_store.set("image_code:%s"%cur_id,text,constants.IMAGE_CODE_REDIS_EXPIRES )
        # 5.判断是否有上个图片验证码编号,有则删除
        if pre_id:
            redis_store.delete("image_code:%s"%pre_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(error=RET.DBERR,errmsg='图片验证码操作异常')
    # 6.返回图片验证码即可
    response=make_response(image_data)
    response.headers["Content-Type"]="image/jpg"
    return response



